home *** CD-ROM | disk | FTP | other *** search
/ Personal Computer World 2009 February / PCWFEB09.iso / Software / Linux / Kubuntu 8.10 / kubuntu-8.10-desktop-i386.iso / casper / filesystem.squashfs / usr / lib / python2.5 / nntplib.pyc (.txt) < prev    next >
Python Compiled Bytecode  |  2008-10-29  |  21KB  |  686 lines

  1. # Source Generated with Decompyle++
  2. # File: in.pyc (Python 2.5)
  3.  
  4. """An NNTP client class based on RFC 977: Network News Transfer Protocol.
  5.  
  6. Example:
  7.  
  8. >>> from nntplib import NNTP
  9. >>> s = NNTP('news')
  10. >>> resp, count, first, last, name = s.group('comp.lang.python')
  11. >>> print 'Group', name, 'has', count, 'articles, range', first, 'to', last
  12. Group comp.lang.python has 51 articles, range 5770 to 5821
  13. >>> resp, subs = s.xhdr('subject', first + '-' + last)
  14. >>> resp = s.quit()
  15. >>>
  16.  
  17. Here 'resp' is the server response line.
  18. Error responses are turned into exceptions.
  19.  
  20. To post an article from a file:
  21. >>> f = open(filename, 'r') # file containing article, including header
  22. >>> resp = s.post(f)
  23. >>>
  24.  
  25. For descriptions of all methods, read the comments in the code below.
  26. Note that all arguments and return values representing article numbers
  27. are strings, not numbers, since they are rarely used for calculations.
  28. """
  29. import re
  30. import socket
  31. __all__ = [
  32.     'NNTP',
  33.     'NNTPReplyError',
  34.     'NNTPTemporaryError',
  35.     'NNTPPermanentError',
  36.     'NNTPProtocolError',
  37.     'NNTPDataError',
  38.     'error_reply',
  39.     'error_temp',
  40.     'error_perm',
  41.     'error_proto',
  42.     'error_data']
  43.  
  44. class NNTPError(Exception):
  45.     '''Base class for all nntplib exceptions'''
  46.     
  47.     def __init__(self, *args):
  48.         Exception.__init__(self, *args)
  49.         
  50.         try:
  51.             self.response = args[0]
  52.         except IndexError:
  53.             self.response = 'No response given'
  54.  
  55.  
  56.  
  57.  
  58. class NNTPReplyError(NNTPError):
  59.     '''Unexpected [123]xx reply'''
  60.     pass
  61.  
  62.  
  63. class NNTPTemporaryError(NNTPError):
  64.     '''4xx errors'''
  65.     pass
  66.  
  67.  
  68. class NNTPPermanentError(NNTPError):
  69.     '''5xx errors'''
  70.     pass
  71.  
  72.  
  73. class NNTPProtocolError(NNTPError):
  74.     '''Response does not begin with [1-5]'''
  75.     pass
  76.  
  77.  
  78. class NNTPDataError(NNTPError):
  79.     '''Error in response data'''
  80.     pass
  81.  
  82. error_reply = NNTPReplyError
  83. error_temp = NNTPTemporaryError
  84. error_perm = NNTPPermanentError
  85. error_proto = NNTPProtocolError
  86. error_data = NNTPDataError
  87. NNTP_PORT = 119
  88. LONGRESP = [
  89.     '100',
  90.     '215',
  91.     '220',
  92.     '221',
  93.     '222',
  94.     '224',
  95.     '230',
  96.     '231',
  97.     '282']
  98. CRLF = '\r\n'
  99.  
  100. class NNTP:
  101.     
  102.     def __init__(self, host, port = NNTP_PORT, user = None, password = None, readermode = None, usenetrc = True):
  103.         """Initialize an instance.  Arguments:
  104.         - host: hostname to connect to
  105.         - port: port to connect to (default the standard NNTP port)
  106.         - user: username to authenticate with
  107.         - password: password to use with username
  108.         - readermode: if true, send 'mode reader' command after
  109.                       connecting.
  110.  
  111.         readermode is sometimes necessary if you are connecting to an
  112.         NNTP server on the local machine and intend to call
  113.         reader-specific comamnds, such as `group'.  If you get
  114.         unexpected NNTPPermanentErrors, you might need to set
  115.         readermode.
  116.         """
  117.         self.host = host
  118.         self.port = port
  119.         self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  120.         self.sock.connect((self.host, self.port))
  121.         self.file = self.sock.makefile('rb')
  122.         self.debugging = 0
  123.         self.welcome = self.getresp()
  124.         readermode_afterauth = 0
  125.         if readermode:
  126.             
  127.             try:
  128.                 self.welcome = self.shortcmd('mode reader')
  129.             except NNTPPermanentError:
  130.                 pass
  131.             except NNTPTemporaryError:
  132.                 e = None
  133.                 if user and e.response[:3] == '480':
  134.                     readermode_afterauth = 1
  135.                 else:
  136.                     raise 
  137.             except:
  138.                 e.response[:3] == '480'
  139.             
  140.  
  141.         None<EXCEPTION MATCH>NNTPPermanentError
  142.         
  143.         try:
  144.             if usenetrc and not user:
  145.                 import netrc as netrc
  146.                 credentials = netrc.netrc()
  147.                 auth = credentials.authenticators(host)
  148.                 if auth:
  149.                     user = auth[0]
  150.                     password = auth[2]
  151.                 
  152.         except IOError:
  153.             pass
  154.  
  155.         if user:
  156.             resp = self.shortcmd('authinfo user ' + user)
  157.             if resp[:3] == '381':
  158.                 if not password:
  159.                     raise NNTPReplyError(resp)
  160.                 else:
  161.                     resp = self.shortcmd('authinfo pass ' + password)
  162.                     if resp[:3] != '281':
  163.                         raise NNTPPermanentError(resp)
  164.                     
  165.             
  166.             if readermode_afterauth:
  167.                 
  168.                 try:
  169.                     self.welcome = self.shortcmd('mode reader')
  170.                 except NNTPPermanentError:
  171.                     pass
  172.                 except:
  173.                     None<EXCEPTION MATCH>NNTPPermanentError
  174.                 
  175.  
  176.             None<EXCEPTION MATCH>NNTPPermanentError
  177.         
  178.  
  179.     
  180.     def getwelcome(self):
  181.         '''Get the welcome message from the server
  182.         (this is read and squirreled away by __init__()).
  183.         If the response code is 200, posting is allowed;
  184.         if it 201, posting is not allowed.'''
  185.         if self.debugging:
  186.             print '*welcome*', repr(self.welcome)
  187.         
  188.         return self.welcome
  189.  
  190.     
  191.     def set_debuglevel(self, level):
  192.         """Set the debugging level.  Argument 'level' means:
  193.         0: no debugging output (default)
  194.         1: print commands and responses but not body text etc.
  195.         2: also print raw lines read and sent before stripping CR/LF"""
  196.         self.debugging = level
  197.  
  198.     debug = set_debuglevel
  199.     
  200.     def putline(self, line):
  201.         '''Internal: send one line to the server, appending CRLF.'''
  202.         line = line + CRLF
  203.         if self.debugging > 1:
  204.             print '*put*', repr(line)
  205.         
  206.         self.sock.sendall(line)
  207.  
  208.     
  209.     def putcmd(self, line):
  210.         '''Internal: send one command to the server (through putline()).'''
  211.         if self.debugging:
  212.             print '*cmd*', repr(line)
  213.         
  214.         self.putline(line)
  215.  
  216.     
  217.     def getline(self):
  218.         '''Internal: return one line from the server, stripping CRLF.
  219.         Raise EOFError if the connection is closed.'''
  220.         line = self.file.readline()
  221.         if self.debugging > 1:
  222.             print '*get*', repr(line)
  223.         
  224.         if not line:
  225.             raise EOFError
  226.         
  227.         if line[-2:] == CRLF:
  228.             line = line[:-2]
  229.         elif line[-1:] in CRLF:
  230.             line = line[:-1]
  231.         
  232.         return line
  233.  
  234.     
  235.     def getresp(self):
  236.         '''Internal: get a response from the server.
  237.         Raise various errors if the response indicates an error.'''
  238.         resp = self.getline()
  239.         if self.debugging:
  240.             print '*resp*', repr(resp)
  241.         
  242.         c = resp[:1]
  243.         if c == '4':
  244.             raise NNTPTemporaryError(resp)
  245.         
  246.         if c == '5':
  247.             raise NNTPPermanentError(resp)
  248.         
  249.         if c not in '123':
  250.             raise NNTPProtocolError(resp)
  251.         
  252.         return resp
  253.  
  254.     
  255.     def getlongresp(self, file = None):
  256.         '''Internal: get a response plus following text from the server.
  257.         Raise various errors if the response indicates an error.'''
  258.         openedFile = None
  259.         
  260.         try:
  261.             if isinstance(file, str):
  262.                 openedFile = file = open(file, 'w')
  263.             
  264.             resp = self.getresp()
  265.             if resp[:3] not in LONGRESP:
  266.                 raise NNTPReplyError(resp)
  267.             
  268.             list = []
  269.             while None:
  270.                 line = self.getline()
  271.                 if line == '.':
  272.                     break
  273.                 
  274.                 if line[:2] == '..':
  275.                     line = line[1:]
  276.                 
  277.                 if file:
  278.                     file.write(line + '\n')
  279.                     continue
  280.                 list.append(line)
  281.             if openedFile:
  282.                 openedFile.close()
  283.             
  284.             return (resp, list)
  285.  
  286.  
  287.     
  288.     def shortcmd(self, line):
  289.         '''Internal: send a command and get the response.'''
  290.         self.putcmd(line)
  291.         return self.getresp()
  292.  
  293.     
  294.     def longcmd(self, line, file = None):
  295.         '''Internal: send a command and get the response plus following text.'''
  296.         self.putcmd(line)
  297.         return self.getlongresp(file)
  298.  
  299.     
  300.     def newgroups(self, date, time, file = None):
  301.         """Process a NEWGROUPS command.  Arguments:
  302.         - date: string 'yymmdd' indicating the date
  303.         - time: string 'hhmmss' indicating the time
  304.         Return:
  305.         - resp: server response if successful
  306.         - list: list of newsgroup names"""
  307.         return self.longcmd('NEWGROUPS ' + date + ' ' + time, file)
  308.  
  309.     
  310.     def newnews(self, group, date, time, file = None):
  311.         """Process a NEWNEWS command.  Arguments:
  312.         - group: group name or '*'
  313.         - date: string 'yymmdd' indicating the date
  314.         - time: string 'hhmmss' indicating the time
  315.         Return:
  316.         - resp: server response if successful
  317.         - list: list of message ids"""
  318.         cmd = 'NEWNEWS ' + group + ' ' + date + ' ' + time
  319.         return self.longcmd(cmd, file)
  320.  
  321.     
  322.     def list(self, file = None):
  323.         '''Process a LIST command.  Return:
  324.         - resp: server response if successful
  325.         - list: list of (group, last, first, flag) (strings)'''
  326.         (resp, list) = self.longcmd('LIST', file)
  327.         for i in range(len(list)):
  328.             list[i] = tuple(list[i].split())
  329.         
  330.         return (resp, list)
  331.  
  332.     
  333.     def description(self, group):
  334.         """Get a description for a single group.  If more than one
  335.         group matches ('group' is a pattern), return the first.  If no
  336.         group matches, return an empty string.
  337.  
  338.         This elides the response code from the server, since it can
  339.         only be '215' or '285' (for xgtitle) anyway.  If the response
  340.         code is needed, use the 'descriptions' method.
  341.  
  342.         NOTE: This neither checks for a wildcard in 'group' nor does
  343.         it check whether the group actually exists."""
  344.         (resp, lines) = self.descriptions(group)
  345.         if len(lines) == 0:
  346.             return ''
  347.         else:
  348.             return lines[0][1]
  349.  
  350.     
  351.     def descriptions(self, group_pattern):
  352.         '''Get descriptions for a range of groups.'''
  353.         line_pat = re.compile('^(?P<group>[^ \t]+)[ \t]+(.*)$')
  354.         (resp, raw_lines) = self.longcmd('LIST NEWSGROUPS ' + group_pattern)
  355.         if resp[:3] != '215':
  356.             (resp, raw_lines) = self.longcmd('XGTITLE ' + group_pattern)
  357.         
  358.         lines = []
  359.         for raw_line in raw_lines:
  360.             match = line_pat.search(raw_line.strip())
  361.             if match:
  362.                 lines.append(match.group(1, 2))
  363.                 continue
  364.         
  365.         return (resp, lines)
  366.  
  367.     
  368.     def group(self, name):
  369.         '''Process a GROUP command.  Argument:
  370.         - group: the group name
  371.         Returns:
  372.         - resp: server response if successful
  373.         - count: number of articles (string)
  374.         - first: first article number (string)
  375.         - last: last article number (string)
  376.         - name: the group name'''
  377.         resp = self.shortcmd('GROUP ' + name)
  378.         if resp[:3] != '211':
  379.             raise NNTPReplyError(resp)
  380.         
  381.         words = resp.split()
  382.         count = first = last = 0
  383.         n = len(words)
  384.         if n > 1:
  385.             count = words[1]
  386.             if n > 2:
  387.                 first = words[2]
  388.                 if n > 3:
  389.                     last = words[3]
  390.                     if n > 4:
  391.                         name = words[4].lower()
  392.                     
  393.                 
  394.             
  395.         
  396.         return (resp, count, first, last, name)
  397.  
  398.     
  399.     def help(self, file = None):
  400.         '''Process a HELP command.  Returns:
  401.         - resp: server response if successful
  402.         - list: list of strings'''
  403.         return self.longcmd('HELP', file)
  404.  
  405.     
  406.     def statparse(self, resp):
  407.         '''Internal: parse the response of a STAT, NEXT or LAST command.'''
  408.         if resp[:2] != '22':
  409.             raise NNTPReplyError(resp)
  410.         
  411.         words = resp.split()
  412.         nr = 0
  413.         id = ''
  414.         n = len(words)
  415.         if n > 1:
  416.             nr = words[1]
  417.             if n > 2:
  418.                 id = words[2]
  419.             
  420.         
  421.         return (resp, nr, id)
  422.  
  423.     
  424.     def statcmd(self, line):
  425.         '''Internal: process a STAT, NEXT or LAST command.'''
  426.         resp = self.shortcmd(line)
  427.         return self.statparse(resp)
  428.  
  429.     
  430.     def stat(self, id):
  431.         '''Process a STAT command.  Argument:
  432.         - id: article number or message id
  433.         Returns:
  434.         - resp: server response if successful
  435.         - nr:   the article number
  436.         - id:   the message id'''
  437.         return self.statcmd('STAT ' + id)
  438.  
  439.     
  440.     def next(self):
  441.         '''Process a NEXT command.  No arguments.  Return as for STAT.'''
  442.         return self.statcmd('NEXT')
  443.  
  444.     
  445.     def last(self):
  446.         '''Process a LAST command.  No arguments.  Return as for STAT.'''
  447.         return self.statcmd('LAST')
  448.  
  449.     
  450.     def artcmd(self, line, file = None):
  451.         '''Internal: process a HEAD, BODY or ARTICLE command.'''
  452.         (resp, list) = self.longcmd(line, file)
  453.         (resp, nr, id) = self.statparse(resp)
  454.         return (resp, nr, id, list)
  455.  
  456.     
  457.     def head(self, id):
  458.         """Process a HEAD command.  Argument:
  459.         - id: article number or message id
  460.         Returns:
  461.         - resp: server response if successful
  462.         - nr: article number
  463.         - id: message id
  464.         - list: the lines of the article's header"""
  465.         return self.artcmd('HEAD ' + id)
  466.  
  467.     
  468.     def body(self, id, file = None):
  469.         """Process a BODY command.  Argument:
  470.         - id: article number or message id
  471.         - file: Filename string or file object to store the article in
  472.         Returns:
  473.         - resp: server response if successful
  474.         - nr: article number
  475.         - id: message id
  476.         - list: the lines of the article's body or an empty list
  477.                 if file was used"""
  478.         return self.artcmd('BODY ' + id, file)
  479.  
  480.     
  481.     def article(self, id):
  482.         '''Process an ARTICLE command.  Argument:
  483.         - id: article number or message id
  484.         Returns:
  485.         - resp: server response if successful
  486.         - nr: article number
  487.         - id: message id
  488.         - list: the lines of the article'''
  489.         return self.artcmd('ARTICLE ' + id)
  490.  
  491.     
  492.     def slave(self):
  493.         '''Process a SLAVE command.  Returns:
  494.         - resp: server response if successful'''
  495.         return self.shortcmd('SLAVE')
  496.  
  497.     
  498.     def xhdr(self, hdr, str, file = None):
  499.         """Process an XHDR command (optional server extension).  Arguments:
  500.         - hdr: the header type (e.g. 'subject')
  501.         - str: an article nr, a message id, or a range nr1-nr2
  502.         Returns:
  503.         - resp: server response if successful
  504.         - list: list of (nr, value) strings"""
  505.         pat = re.compile('^([0-9]+) ?(.*)\n?')
  506.         (resp, lines) = self.longcmd('XHDR ' + hdr + ' ' + str, file)
  507.         for i in range(len(lines)):
  508.             line = lines[i]
  509.             m = pat.match(line)
  510.             if m:
  511.                 lines[i] = m.group(1, 2)
  512.                 continue
  513.         
  514.         return (resp, lines)
  515.  
  516.     
  517.     def xover(self, start, end, file = None):
  518.         '''Process an XOVER command (optional server extension) Arguments:
  519.         - start: start of range
  520.         - end: end of range
  521.         Returns:
  522.         - resp: server response if successful
  523.         - list: list of (art-nr, subject, poster, date,
  524.                          id, references, size, lines)'''
  525.         (resp, lines) = self.longcmd('XOVER ' + start + '-' + end, file)
  526.         xover_lines = []
  527.         for line in lines:
  528.             elem = line.split('\t')
  529.             
  530.             try:
  531.                 xover_lines.append((elem[0], elem[1], elem[2], elem[3], elem[4], elem[5].split(), elem[6], elem[7]))
  532.             continue
  533.             except IndexError:
  534.                 raise NNTPDataError(line)
  535.                 continue
  536.             
  537.  
  538.         
  539.         return (resp, xover_lines)
  540.  
  541.     
  542.     def xgtitle(self, group, file = None):
  543.         '''Process an XGTITLE command (optional server extension) Arguments:
  544.         - group: group name wildcard (i.e. news.*)
  545.         Returns:
  546.         - resp: server response if successful
  547.         - list: list of (name,title) strings'''
  548.         line_pat = re.compile('^([^ \t]+)[ \t]+(.*)$')
  549.         (resp, raw_lines) = self.longcmd('XGTITLE ' + group, file)
  550.         lines = []
  551.         for raw_line in raw_lines:
  552.             match = line_pat.search(raw_line.strip())
  553.             if match:
  554.                 lines.append(match.group(1, 2))
  555.                 continue
  556.         
  557.         return (resp, lines)
  558.  
  559.     
  560.     def xpath(self, id):
  561.         '''Process an XPATH command (optional server extension) Arguments:
  562.         - id: Message id of article
  563.         Returns:
  564.         resp: server response if successful
  565.         path: directory path to article'''
  566.         resp = self.shortcmd('XPATH ' + id)
  567.         if resp[:3] != '223':
  568.             raise NNTPReplyError(resp)
  569.         
  570.         
  571.         try:
  572.             (resp_num, path) = resp.split()
  573.         except ValueError:
  574.             raise NNTPReplyError(resp)
  575.  
  576.         return (resp, path)
  577.  
  578.     
  579.     def date(self):
  580.         '''Process the DATE command. Arguments:
  581.         None
  582.         Returns:
  583.         resp: server response if successful
  584.         date: Date suitable for newnews/newgroups commands etc.
  585.         time: Time suitable for newnews/newgroups commands etc.'''
  586.         resp = self.shortcmd('DATE')
  587.         if resp[:3] != '111':
  588.             raise NNTPReplyError(resp)
  589.         
  590.         elem = resp.split()
  591.         if len(elem) != 2:
  592.             raise NNTPDataError(resp)
  593.         
  594.         date = elem[1][2:8]
  595.         time = elem[1][-6:]
  596.         if len(date) != 6 or len(time) != 6:
  597.             raise NNTPDataError(resp)
  598.         
  599.         return (resp, date, time)
  600.  
  601.     
  602.     def post(self, f):
  603.         '''Process a POST command.  Arguments:
  604.         - f: file containing the article
  605.         Returns:
  606.         - resp: server response if successful'''
  607.         resp = self.shortcmd('POST')
  608.         if resp[0] != '3':
  609.             raise NNTPReplyError(resp)
  610.         
  611.         while None:
  612.             line = f.readline()
  613.             if not line:
  614.                 break
  615.             
  616.             if line[-1] == '\n':
  617.                 line = line[:-1]
  618.             
  619.             if line[:1] == '.':
  620.                 line = '.' + line
  621.             
  622.             continue
  623.             self.putline('.')
  624.             return self.getresp()
  625.  
  626.     
  627.     def ihave(self, id, f):
  628.         '''Process an IHAVE command.  Arguments:
  629.         - id: message-id of the article
  630.         - f:  file containing the article
  631.         Returns:
  632.         - resp: server response if successful
  633.         Note that if the server refuses the article an exception is raised.'''
  634.         resp = self.shortcmd('IHAVE ' + id)
  635.         if resp[0] != '3':
  636.             raise NNTPReplyError(resp)
  637.         
  638.         while None:
  639.             line = f.readline()
  640.             if not line:
  641.                 break
  642.             
  643.             if line[-1] == '\n':
  644.                 line = line[:-1]
  645.             
  646.             if line[:1] == '.':
  647.                 line = '.' + line
  648.             
  649.             continue
  650.             self.putline('.')
  651.             return self.getresp()
  652.  
  653.     
  654.     def quit(self):
  655.         '''Process a QUIT command and close the socket.  Returns:
  656.         - resp: server response if successful'''
  657.         resp = self.shortcmd('QUIT')
  658.         self.file.close()
  659.         self.sock.close()
  660.         del self.file
  661.         del self.sock
  662.         return resp
  663.  
  664.  
  665. if __name__ == '__main__':
  666.     import os
  667.     if 'news':
  668.         pass
  669.     newshost = os.environ['NNTPSERVER']
  670.     if newshost.find('.') == -1:
  671.         mode = 'readermode'
  672.     else:
  673.         mode = None
  674.     s = NNTP(newshost, readermode = mode)
  675.     (resp, count, first, last, name) = s.group('comp.lang.python')
  676.     print resp
  677.     print 'Group', name, 'has', count, 'articles, range', first, 'to', last
  678.     (resp, subs) = s.xhdr('subject', first + '-' + last)
  679.     print resp
  680.     for item in subs:
  681.         print '%7s %s' % item
  682.     
  683.     resp = s.quit()
  684.     print resp
  685.  
  686.